在pegmatite-clocks.dtsi中audio device的clocks如下

```
1.
           ref clk25mhz: clk25mhz {
 2.
                compatible = "fixed-clock";
 3.
                #clock-cells = <0>;
 4.
                clock-frequency = <25000000>;
 5.
           };
 6.
 7.
            . . . . . .
 8.
           system_pll: systempll {
 9.
10.
                compatible = "marvell,pegmatite-pll";
11.
                #clock-cells = <0>;
12.
                reg = \langle 0 \ 0xd0621800 \ 0 \ 0x200 \rangle;
13.
                clocks = <&ref_clk25mhz>;
14.
                clock-frequency = <2500000000>;
15.
           };
16.
17.
           system_pll_gate: systempllgate {
18.
                compatible = "marvell,pegmatite-clkgate";
19.
                #clock-cells = <0>;
20.
                reg = \langle 0 \ 0xd0627018 \ 0 \ 0x8 \rangle;
21.
                clocks = <&system_pll>;
22.
                always-used;
23.
           };
24.
25.
26.
27.
           preaudio_clk: preaudioclk {
28.
                compatible = "marvell,pegmatite-clkgen";
29.
                #clock-cells = <0>;
30.
                reg = \langle 0 \ 0xf9080100 \ 0 \ 0x8 \rangle;
31.
                prediv-shift = <25>;
32.
                clocks = <&system_pll_gate>, <&ref_clk25mhz>;
33.
                clock-source = <0>;
34.
                max-divide = <30>;
35.
                clock-frequency = <83333333;</pre>
36.
           };
37.
38.
           audio clk: audioclk {
39.
                compatible = "marvell,pegmatite-clkgen";
40.
                #clock-cells = <0>;
41.
                reg = \langle 0 \ 0xf9080108 \ 0 \ 0x8 \rangle;
42.
                clocks = <&preaudio_clk>;
43.
                max-divide = <60>;
44.
                clock-frequency = <1536000>;
45.
           };
46.
47.
           audio_clkgate: audioclkgate {
48.
                compatible = "marvell,pegmatite-clkgate";
49.
                #clock-cells = <0>;
50.
                reg = \langle 0 \ 0xf9080108 \ 0 \ 0x8 \rangle;
51.
                clocks = <&audio clk>;
52.
           };
```

从最源头的clock到audio device接收到的clock共经过了6层clock。

1

这可能是个25MHz的晶振,提供固定频率

2

这是被PLL(锁相环)提高了100倍频率的system clock,ARM core应该在此频率下工作

(3)

提供对system clock的gate功能,但由于 always-used; property,所以连接到该clock实际 上没有关断功能。

**(4)** 

这是audio device的pre-divisor clock。它接受的是2.5GHz的频率,经过分频后希望输出的频率是

```
clock-frequency = <833333333>;
max-divide = <30>;
为什么是30?
```

该clock的输入是2.5GHz,输出希望是83333333.

2500000000 / 83333333 = 30

即divisor = 30

按照Pre-divisor clock config register的定义,该divisor要被拆分成hidiv和lodiv.

divisor = hidiv + lowdiv

in drivers/clk/pegmatite/clkgen.c

```
1.
          if (gen->use_div_select) {
                                        (D)
 2.
              u32 div_sel = 0;
 3.
 4.
              if ((parent_rate / 2) > rate)
 5.
                  div_sel = 1;
 6.
 7.
              val \&= \sim (1 << DIV SEL SHIFT);
8.
              val |= (div_sel << DIV_SEL_SHIFT);</pre>
9.
          }
10.
          else {
                                         (E)
11.
              unsigned int hidiv = 0, lodiv = 0, prediv = 0;
12.
13.
14.
               * If the parent_rate (w/ or w/o predivider) matches the requested r
      ate
15.
               * then no further dividers are needed
16.
               */
17.
              if (parent_rate > rate) { (F)
18.
                  unsigned int div;
19.
20.
                  /*
                   * If the max hi-lo divide cannot get the clock rate slow enough
21.
                   * use the predivider
22.
23.
24.
                  if (gen->use_prediv) { (G)
25.
                      if ((parent_rate / gen->max_divide) > rate) {
                                                                        (H)
26.
                           prediv = readl(gen->config + 4);
                                                                         (I)
27.
                           prediv >>= gen->prediv shift;
                                                                         (J)
28.
                          prediv &= PRE_DIV_VAL_MASK;
29.
30.
                          if (prediv)
31.
                               parent_rate /= prediv;
                                                                          (K)
32.
                      }
33.
                  }
34.
                  /*
                   * Caclulate the divisor required to divide the parent rate (w/
35.
      or w/o predivisor) down to the requested rate
36.
                   */
                  if (parent_rate > rate) {
37.
38.
                      div = parent rate / rate;
                                                     (A)
39.
                      if (abs(rate - parent_rate / div) > abs(rate - parent_rate /
       (div + 1))
                      (B)
40.
                          ++div;
41.
42.
                      if (div > gen->max_divide) {
43.
                          div = gen->max_divide;
44.
                          pr_err("%s: %s divisor %d greater than max %d!!\n", __fu
      nc__, hw->init->name, div, gen->max_divide);
45.
46.
47.
                      hidiv = div / 2;
                                               (C)
48.
                      lodiv = div - hidiv;
```

parent\_rate = 2.5GHz (input)

rate = 83333333 (output)

(D)

只有DDR clock用到了 use\_div\_select property

(E)

audio clock应该走该branch

(F)

自然满足该条件(2.5G > 83333333)

(G)

audio使用了 prediv

(H)

2.5G / 30 = 83333333 > 83333333

不满足

但从实际效果上看(I)(J)(K)的code还是执行了!!!

(I)(J)

内置prediv的值记录在 AudioClk Pre-divider ClkGen Configuration Register bit25 to bit31.

从0xF9080104读出是0x04080704,则该ASIC内置的prediv = 2。

(K)

parent rate = 2.5G / 2 = 1.25G

- (A) div = 30 (在parent\_rate = 2.5G的情况下), div = 15(在parent\_rate = 1.25G的情况下)
- (B) 由于div是整数,舍弃了小数,所以这里的判断实际上是看div应该往那个方向舍入(四舍 五入),这里应该是舍去小数,所以div不需要+1
- (C) hidiv = 30 / 2 = 15, lodiv = 30 15 = 15 (在parent\_rate = 2.5G的情况下)

hidiv = 15 / 2 = 8, lodiv = 15 - 8 = 7 (在parent rate = 1.25G的情况下)

| AudioClk Pre-divider ClkGen Configuration | value      |
|-------------------------------------------|------------|
| 0xF908,0100                               | 0x00080704 |

即HIDIV = 08,LODIV = 07

从实际效果看, system\_pll的clock 必然>= 2583333323,而非精确的25000000。

(5)

AudioClk clock config register的分频设置

输入是接近83333333(83M)Hz的频率(应该不是精确的83333333, 因为pre-divisor clock的 divisor是整数, 有舍入误差)

输出希望是1536000(1.5M)

83333333 / 1536000 = 54 (54.253472005)

## 这个divisor是小于下面的设置的

max-divide = <60>;

hidiv = 54 / 2 = 27

lodiv = 54 - 27 = 27

| AudioClk ClkGen Configuration Register | value      |
|----------------------------------------|------------|
| 0xF908,0108                            | 0x001B1B02 |

即HIDIV = 0x1B (27), LODIV = 0x1B (27) 这个手工计算的与dump register是一致的!

6

audioclkgate clock指示提供gate functionality。在low power时可以关断该clock。